/*
 * Copyright 2014-2023 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package de.codecentric.boot.admin.server.services;

import java.time.Duration;
import java.util.logging.Level;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.test.StepVerifier;

import de.codecentric.boot.admin.server.domain.entities.EventsourcingInstanceRepository;
import de.codecentric.boot.admin.server.domain.entities.Instance;
import de.codecentric.boot.admin.server.domain.entities.InstanceRepository;
import de.codecentric.boot.admin.server.domain.events.InstanceEndpointsDetectedEvent;
import de.codecentric.boot.admin.server.domain.values.Endpoints;
import de.codecentric.boot.admin.server.domain.values.InstanceId;
import de.codecentric.boot.admin.server.domain.values.Registration;
import de.codecentric.boot.admin.server.domain.values.StatusInfo;
import de.codecentric.boot.admin.server.eventstore.ConcurrentMapEventStore;
import de.codecentric.boot.admin.server.eventstore.InMemoryEventStore;
import de.codecentric.boot.admin.server.services.endpoints.EndpointDetectionStrategy;

import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

class EndpointDetectorTest {

	private EndpointDetector detector;

	private InstanceRepository repository;

	private ConcurrentMapEventStore eventStore;

	private EndpointDetectionStrategy strategy;

	@BeforeEach
	void setup() {
		eventStore = new InMemoryEventStore();
		repository = new EventsourcingInstanceRepository(eventStore);
		strategy = mock(EndpointDetectionStrategy.class);
		detector = new EndpointDetector(repository, strategy);
	}

	@Test
	void should_update_endpoints() {
		// given
		Registration registration = Registration.create("foo", "https://health").managementUrl("https://mgmt").build();
		Instance instance = Instance.create(InstanceId.of("onl"))
			.register(registration)
			.withStatusInfo(StatusInfo.ofUp());
		StepVerifier.create(repository.save(instance)).expectNextCount(1).verifyComplete();

		Instance noActuator = Instance.create(InstanceId.of("noActuator"))
			.register(Registration.create("foo", "https://health").build())
			.withStatusInfo(StatusInfo.ofUp());
		StepVerifier.create(repository.save(noActuator)).expectNextCount(1).verifyComplete();

		Instance offline = Instance.create(InstanceId.of("off"))
			.register(registration)
			.withStatusInfo(StatusInfo.ofOffline());
		StepVerifier.create(repository.save(offline)).expectNextCount(1).verifyComplete();

		Instance unknown = Instance.create(InstanceId.of("unk"))
			.register(registration)
			.withStatusInfo(StatusInfo.ofUnknown());
		StepVerifier.create(repository.save(unknown)).expectNextCount(1).verifyComplete();

		when(strategy.detectEndpoints(any(Instance.class))).thenReturn(Mono.just(Endpoints.single("id", "url")));

		// when/then
		StepVerifier.create(Flux.from(eventStore).log("FOO", Level.SEVERE))
			.expectSubscription()
			.then(() -> StepVerifier.create(detector.detectEndpoints(offline.getId())).verifyComplete())
			.then(() -> StepVerifier.create(detector.detectEndpoints(unknown.getId())).verifyComplete())
			.then(() -> StepVerifier.create(detector.detectEndpoints(noActuator.getId())).verifyComplete())
			.expectNoEvent(Duration.ofMillis(100L))
			.then(() -> StepVerifier.create(detector.detectEndpoints(instance.getId())).verifyComplete())
			.assertNext((event) -> assertThat(event).isInstanceOf(InstanceEndpointsDetectedEvent.class))
			.thenCancel()
			.verify();

		StepVerifier.create(repository.find(instance.getId()))
			.assertNext((app) -> assertThat(app.getEndpoints())
				.isEqualTo(Endpoints.single("id", "url").withEndpoint("health", "https://health")))
			.verifyComplete();
	}

}
